home *** CD-ROM | disk | FTP | other *** search
/ Celestin Apprentice 5 / Apprentice-Release5.iso / Source Code / C / Snippets / MacLZSS 1.0.3 / lzss.c < prev    next >
Text File  |  1996-07-07  |  9KB  |  265 lines

  1. /* ----------------------------------------------------------------------
  2.  
  3.     MacLZSS - a data compression program
  4.     version 1.0.2
  5.     
  6.     Written by: Haruhiko Okumura
  7.     Modifications by: Rob Elliott
  8.     Ported to CW by: Paul Celestin
  9.     
  10.     950630 - 1.0.0 - initial port to CW
  11.     951215 - 1.0.1 - updated for CW7
  12.     960704 - 1.0.2 - updated for CW9
  13.  
  14. ---------------------------------------------------------------------- */
  15.  
  16. /**************************************************************
  17.     Use, distribute, and modify this program freely.
  18.     Please send me your improved versions.
  19.         PC-VAN        SCIENCE
  20.         NIFTY-Serve    PAF01022
  21.         CompuServe    74050,1022
  22.     
  23.     ---    
  24.     26 May 1989 Modifications by Rob Elliott
  25.     Usenet     rob@embossed.com
  26.     FidoNet    1:115/729
  27.     CompuServe 70675,1204
  28.     
  29.     Converted to AT&T System V UNIX, then Macintosh THINK 'C'
  30.     
  31.     - Changed function prototypes into old-style declarations,
  32.     e.g.         void DecodeChar(int x)
  33.     becomes     void DecodeChar(x) int x;
  34.     - changed 'int' to 'short', 'unsigned' to 'long', etc. as
  35.     needed (system & compiler dependent)
  36.     
  37. **************************************************************/
  38.  
  39. #include <stdio.h>
  40. /* #include <stdlib.h>   not needed */
  41. #include <strings.h>    /* was string.h */
  42. #include <ctype.h>
  43. #define N         4096    /* size of ring buffer */
  44. #define F           18    /* upper limit for match_length */
  45. #define THRESHOLD    2   /* encode string into position and length
  46.                            if match_length is greater than this */
  47. #define NIL            N    /* index for root of binary search trees */
  48. #define EXIT_SUCCESS 0
  49. #define EXIT_FAILURE 1
  50. /* #define int short    needed for some UNIX versions */
  51.  
  52. unsigned long 
  53.         textsize = 0,    /* text size counter */
  54.         codesize = 0,    /* code size counter */
  55.         printcount = 0;    /* counter for reporting progress every 1K bytes */
  56. unsigned char
  57.         text_buf[N + F - 1];    /* ring buffer of size N,
  58.             with extra F-1 bytes to facilitate string comparison */
  59. int        match_position, match_length,  /* of longest match.  These are
  60.             set by the InsertNode() procedure. */
  61.         lson[N + 1], rson[N + 257], dad[N + 1];  /* left & right children &
  62.             parents -- These constitute binary search trees. */
  63. FILE    *infile, *outfile;  /* input & output files */
  64. void InitTree()  /* initialize trees */
  65.  
  66. {
  67.     int  i;
  68.     /* For i = 0 to N - 1, rson[i] and lson[i] will be the right and
  69.        left children of node i.  These nodes need not be initialized.
  70.        Also, dad[i] is the parent of node i.  These are initialized to
  71.        NIL (= N), which stands for 'not used.'
  72.        For i = 0 to 255, rson[N + i + 1] is the root of the tree
  73.        for strings that begin with character i.  These are initialized
  74.        to NIL.  Note there are 256 trees. */
  75.     for (i = N + 1; i <= N + 256; i++) rson[i] = NIL;
  76.     for (i = 0; i < N; i++) dad[i] = NIL;
  77. }
  78. void InsertNode(r)
  79. int r;
  80.     /* Inserts string of length F, text_buf[r..r+F-1], into one of the
  81.        trees (text_buf[r]'th tree) and returns the longest-match position
  82.        and length via the global variables match_position and match_length.
  83.        If match_length = F, then removes the old node in favor of the new
  84.        one, because the old one will be deleted sooner.
  85.        Note r plays double role, as tree node and position in buffer. */
  86. {
  87.     int  i, p, cmp;
  88.     unsigned char  *key;
  89.     cmp = 1;  key = &text_buf[r];  p = N + 1 + key[0];
  90.     rson[r] = lson[r] = NIL;  match_length = 0;
  91.     for ( ; ; ) {
  92.         if (cmp >= 0) {
  93.             if (rson[p] != NIL) p = rson[p];
  94.             else {  rson[p] = r;  dad[r] = p;  return;  }
  95.         } else {
  96.             if (lson[p] != NIL) p = lson[p];
  97.             else {  lson[p] = r;  dad[r] = p;  return;  }
  98.         }
  99.         for (i = 1; i < F; i++)
  100.             if ((cmp = key[i] - text_buf[p + i]) != 0)  break;
  101.         if (i > match_length) {
  102.             match_position = p;
  103.             if ((match_length = i) >= F)  break;
  104.         }
  105.     }
  106.     dad[r] = dad[p];  lson[r] = lson[p];  rson[r] = rson[p];
  107.     dad[lson[p]] = r;  dad[rson[p]] = r;
  108.     if (rson[dad[p]] == p) rson[dad[p]] = r;
  109.     else                   lson[dad[p]] = r;
  110.     dad[p] = NIL;  /* remove p */
  111. }
  112. void DeleteNode(p)  /* deletes node p from tree */
  113. int p;
  114. {
  115.     int  q;
  116.     
  117.     if (dad[p] == NIL) return;  /* not in tree */
  118.     if (rson[p] == NIL) q = lson[p];
  119.     else if (lson[p] == NIL) q = rson[p];
  120.     else {
  121.         q = lson[p];
  122.         if (rson[q] != NIL) {
  123.             do {  q = rson[q];  } while (rson[q] != NIL);
  124.             rson[dad[q]] = lson[q];  dad[lson[q]] = dad[q];
  125.             lson[q] = lson[p];  dad[lson[p]] = q;
  126.         }
  127.         rson[q] = rson[p];  dad[rson[p]] = q;
  128.     }
  129.     dad[q] = dad[p];
  130.     if (rson[dad[p]] == p) rson[dad[p]] = q;  else lson[dad[p]] = q;
  131.     dad[p] = NIL;
  132. }
  133. void Encode()
  134.  
  135. {
  136.     int  i, c, len, r, s, last_match_length, code_buf_ptr;
  137.     unsigned char  code_buf[17], mask;
  138.     
  139.     InitTree();  /* initialize trees */
  140.     code_buf[0] = 0;  /* code_buf[1..16] saves eight units of code, and
  141.         code_buf[0] works as eight flags, "1" representing that the unit
  142.         is an unencoded letter (1 byte), "0" a position-and-length pair
  143.         (2 bytes).  Thus, eight units require at most 16 bytes of code. */
  144.     code_buf_ptr = mask = 1;
  145.     s = 0;  r = N - F;
  146.     for (i = s; i < r; i++) text_buf[i] = ' ';  /* Clear the buffer with
  147.         any character that will appear often. */
  148.     for (len = 0; len < F && (c = getc(infile)) != EOF; len++)
  149.         text_buf[r + len] = c;  /* Read F bytes into the last F bytes of
  150.             the buffer */
  151.     if ((textsize = len) == 0) return;  /* text of size zero */
  152.     for (i = 1; i <= F; i++) InsertNode(r - i);  /* Insert the F strings,
  153.         each of which begins with one or more 'space' characters.  Note
  154.         the order in which these strings are inserted.  This way,
  155.         degenerate trees will be less likely to occur. */
  156.     InsertNode(r);  /* Finally, insert the whole string just read.  The
  157.         global variables match_length and match_position are set. */
  158.     do {
  159.         if (match_length > len) match_length = len;  /* match_length
  160.             may be spuriously long near the end of text. */
  161.         if (match_length <= THRESHOLD) {
  162.             match_length = 1;  /* Not long enough match.  Send one byte. */
  163.             code_buf[0] |= mask;  /* 'send one byte' flag */
  164.             code_buf[code_buf_ptr++] = text_buf[r];  /* Send uncoded. */
  165.         } else {
  166.             code_buf[code_buf_ptr++] = (unsigned char) match_position;
  167.             code_buf[code_buf_ptr++] = (unsigned char)
  168.                 (((match_position >> 4) & 0xf0)
  169.               | (match_length - (THRESHOLD + 1)));  /* Send position and
  170.                     length pair. Note match_length > THRESHOLD. */
  171.         }
  172.         if ((mask <<= 1) == 0) {  /* Shift mask left one bit. */
  173.             for (i = 0; i < code_buf_ptr; i++)  /* Send at most 8 units of */
  174.                 putc(code_buf[i], outfile);     /* code together */
  175.             codesize += code_buf_ptr;
  176.             code_buf[0] = 0;  code_buf_ptr = mask = 1;
  177.         }
  178.         last_match_length = match_length;
  179.         for (i = 0; i < last_match_length &&
  180.                 (c = getc(infile)) != EOF; i++) {
  181.             DeleteNode(s);        /* Delete old strings and */
  182.             text_buf[s] = c;    /* read new bytes */
  183.             if (s < F - 1) text_buf[s + N] = c;  /* If the position is
  184.                 near the end of buffer, extend the buffer to make
  185.                 string comparison easier. */
  186.             s = (s + 1) & (N - 1);  r = (r + 1) & (N - 1);
  187.                 /* Since this is a ring buffer, increment the position
  188.                    modulo N. */
  189.             InsertNode(r);    /* Register the string in text_buf[r..r+F-1] */
  190.         }
  191.         if ((textsize += i) > printcount) {
  192.             printf("%12ld\r", textsize);  printcount += 1024;
  193.                 /* Reports progress each time the textsize exceeds
  194.                    multiples of 1024. */
  195.         }
  196.         while (i++ < last_match_length) {    /* After the end of text, */
  197.             DeleteNode(s);                    /* no need to read, but */
  198.             s = (s + 1) & (N - 1);  r = (r + 1) & (N - 1);
  199.             if (--len) InsertNode(r);        /* buffer may not be empty. */
  200.         }
  201.     } while (len > 0);    /* until length of string to be processed is zero */
  202.     if (code_buf_ptr > 1) {        /* Send remaining code. */
  203.         for (i = 0; i < code_buf_ptr; i++) putc(code_buf[i], outfile);
  204.         codesize += code_buf_ptr;
  205.     }
  206.     printf("In : %ld bytes\n", textsize);    /* Encoding is done. */
  207.     printf("Out: %ld bytes\n", codesize);
  208.     printf("Out/In: %.3f\n", (double)codesize / textsize);
  209. }
  210. void Decode()    /* Just the reverse of Encode(). */
  211.  
  212. {
  213.     int  i, j, k, r, c;
  214.     unsigned int  flags;
  215.     
  216.     for (i = 0; i < N - F; i++) text_buf[i] = ' ';
  217.     r = N - F;  flags = 0;
  218.     for ( ; ; ) {
  219.         if (((flags >>= 1) & 256) == 0) {
  220.             if ((c = getc(infile)) == EOF) break;
  221.             flags = c | 0xff00;        /* uses higher byte cleverly */
  222.         }                            /* to count eight */
  223.         if (flags & 1) {
  224.             if ((c = getc(infile)) == EOF) break;
  225.             putc(c, outfile);  text_buf[r++] = c;  r &= (N - 1);
  226.         } else {
  227.             if ((i = getc(infile)) == EOF) break;
  228.             if ((j = getc(infile)) == EOF) break;
  229.             i |= ((j & 0xf0) << 4);  j = (j & 0x0f) + THRESHOLD;
  230.             for (k = 0; k <= j; k++) {
  231.                 c = text_buf[(i + k) & (N - 1)];
  232.                 putc(c, outfile);  text_buf[r++] = c;  r &= (N - 1);
  233.             }
  234.         }
  235.     }
  236. }
  237. int _main(argc, argv)
  238. int argc;
  239. char *argv[];
  240. {
  241.     char  *s;
  242.     
  243. /*    infile = stdin;                For filter operation
  244.     outfile = stdout;
  245.     if (argc > 1) {
  246.         printf("Decoding...\n");
  247.         Decode(); }
  248.     else
  249.         Encode();*/
  250.  
  251.     if (argc != 4) {
  252.         printf("'lzss e file1 file2' encodes file1 into file2.\n"
  253.                "'lzss d file2 file1' decodes file2 into file1.\n");
  254.         return EXIT_FAILURE;
  255.     }
  256.     if ((s = argv[1], s[1] || strpbrk(s, "DEde") == NULL)
  257.      || (s = argv[2], (infile  = fopen(s, "rb")) == NULL)
  258.      || (s = argv[3], (outfile = fopen(s, "wb")) == NULL)) {
  259.         printf("??? %s\n", s);  return EXIT_FAILURE;
  260.     }
  261.     if (toupper(*argv[1]) == 'E') Encode();  else Decode();
  262.     fclose(infile);  fclose(outfile);
  263.     return EXIT_SUCCESS;
  264. }
  265.